# **RPicSim**

David Grayson
Las Vegas Ruby Meetup
2014-02-12

https://github.com/pololu/rpicsim

The RPicSim gem provides an interface to the MPLAB X PIC simulator that allows you to write simulator-based automated tests of PIC firmware using Ruby and RSpec.



#### PIC microcontrollers



#### PIC development tools



**RPicSim** 

### PIC microcontrollers









## PIC development tools



# Debugging firmware

- Oscilloscope
- Dummy code to toggle I/O lines



#### MPLAB X Simulator



# **RPicSim**



The RPicSim gem provides an interface to the MPLAB X PIC simulator that allows you to write simulator-based automated tests of PIC firmware using Ruby and RSpec.

#### MPLAB X IDE

▼ 🔐 ▼ 🛂 ▼ 🛂 ▼ 🔐 ▼ PC: 0x0 | z dc c : W:0x0 : bank 0 | Q | Search (Ctrl+t)

MPLAB X IDE v2.00 - morse\_sender : default

File Edit View Navigate Source Refactor Run Debug Team Tools Window Help

😷 🚰 🛂 🤚 🍏 🍘 default



MPLAB X Simulator (jar files)

## Simulating I/O

```
it "continuously mirrors" do
   main_input.set false
   run_cycles 10
   expect(main_output).to be_driving_low
   main_input.set true
   run_cycles 10
   expect(main_output).to be_driving_high
end
```

## Persistent expectations

```
expect(main_output).to be_driving_low
```

```
expecting main_output => be_driving_low
```

#### Unit test

```
it 'adds 70 to 22' do
  addend1.value = 70
  addend2.value = 22
  run_subroutine :addition, cycle_limit: 100
  expect(sum.value).to eq 92
end
```

#### Unit test with RAM watcher

```
it 'adds 70 to 22' do
  addend1.value = 70
  addend2.value = 22
  step; ram_watcher.clear
  run_subroutine :addition, cycle_limit: 100
  expect(sim.ram_watcher.writes).to eq({sum: 92})
end
```

### Setup

```
require 'rpicsim/rspec'
class MySim < RPicSim::Sim</pre>
  device_is 'PIC10F322'
  filename_is File.dirname(__FILE__) + '.../src/dist/firmware.cof'
  # pin aliases
  # variable definitions
  # helper methods
end
describe 'some part of the firmware' do
  before do
    start_sim MySim
  end
  # examples
end
```

#### Stubs

#### Failures:

```
1) FooWidget when exposed to 1.5 ms pulses behaves correctly
  Failure/Error: run microseconds 1500
    expected INTCON to satisfy block
  # ./lib/rpicsim/rspec/persistent expectations.rb:29:in `check expectations'
  # ./lib/rpicsim/rspec/persistent expectations.rb:27:in `check expectations'
  # ./lib/rpicsim/rspec/helpers.rb:25:in `start sim'
  # ./lib/rpicsim/sim.rb:574:in `step'
  # ./lib/rpicsim/sim.rb:716:in `run to cycle count'
  # ./lib/rpicsim/sim.rb:708:in `run cycles'
  # ./spec/foo widget spec.rb:10:in `(root)'
  Simulation cycle count: 78963
                                                 Useful error
  Simulation stack trace:
  0 \times 01A0 = startMotor
  0x0044 = motorService + 0x14
                                                   messages
  0x0B12 = mainLoop+0x2
  0x008C = start2
```

Finished in 4.55 seconds 44 examples, 1 failure

Failed examples:

rspec ./spec/example/nice\_error\_spec.rb:8 # FooWidget when exposed to 1.5ms pulses behaves correctly



#### PIC microcontrollers



PIC development tools



**RPicSim** 

For more info: https://github.com/pololu/rpicsim

TABLE 23-2: PIC10(L)F320/322 INSTRUCTION SET

| Mnemonic,<br>Operands                  |      | Description                  | Cycles   | 14-Bit Opcode |      |      | Status | Notes    |         |  |
|----------------------------------------|------|------------------------------|----------|---------------|------|------|--------|----------|---------|--|
|                                        |      | Description                  |          | MSb           |      |      | LSb    | Affected | Notes   |  |
| BYTE-ORIENTED FILE REGISTER OPERATIONS |      |                              |          |               |      |      |        |          |         |  |
| ADDWF                                  | f, d | Add W and f                  | 1        | 00            | 0111 | dfff | ffff   | C, DC, Z | 1, 2    |  |
| ANDWF                                  | f, d | AND W with f                 | 1        | 00            | 0101 | dfff | ffff   | Z 1, 2   |         |  |
| CLRF                                   | f    | Clear f                      | 1        | 00            | 0001 | lfff | ffff   | Z        | 2       |  |
| CLRW                                   | _    | Clear W                      | 1        | 00            | 0001 | 0xxx | xxxx   | Z        |         |  |
| COMF                                   | f, d | Complement f                 | 1        | 00            | 1001 | dfff | ffff   | Z 1, 2   |         |  |
| DECF                                   | f, d | Decrement f                  | 1        | 00            | 0011 | dfff | ffff   | Z        | 1, 2    |  |
| DECFSZ                                 | f, d | Decrement f, Skip if 0       | 1(2)     | 00            | 1011 | dfff | ffff   |          | 1, 2, 3 |  |
| INCF                                   | f, d | Increment f                  | 1        | 00            | 1010 | dfff | ffff   | Z        | 1, 2    |  |
| INCFSZ                                 | f, d | Increment f, Skip if 0       | 1(2)     | 00            | 1111 | dfff | ffff   |          | 1, 2, 3 |  |
| IORWF                                  | f, d | Inclusive OR W with f        | 1        | 00            | 0100 | dfff | ffff   | Z        | 1, 2    |  |
| MOVF                                   | f, d | Move f                       | 1        | 00            | 1000 | dfff | ffff   | Z        | 1, 2    |  |
| MOVWF                                  | f    | Move W to f                  | 1        | 00            | 0000 | lfff | ffff   |          |         |  |
| NOP                                    | _    | No Operation                 | 1        | 00            | 0000 | 0xx0 | 0000   |          |         |  |
| RLF                                    | f, d | Rotate Left f through Carry  | 1        | 00            | 1101 | dfff | ffff   | С        | 1, 2    |  |
| RRF                                    | f, d | Rotate Right f through Carry | 1        | 00            | 1100 | dfff | ffff   | С        | 1, 2    |  |
| SUBWF                                  | f, d | Subtract W from f            | 1        | 00            | 0010 | dfff | ffff   | C, DC, Z | 1, 2    |  |
| SWAPF                                  | f, d | Swap nibbles in f            | 1        | 00            | 1110 | dfff | ffff   |          | 1, 2    |  |
| XORWF                                  | f, d | Exclusive OR W with f        | 1        | 00            | 0110 | dfff | ffff   | Z        | 1, 2    |  |
|                                        |      | BIT-ORIENTED FILE REGIS      | TER OPER | RATION        | IS   |      |        |          |         |  |
| BCF                                    | f, b | Bit Clear f                  | 1        | 01            | 00bb | bfff | ffff   |          | 1, 2    |  |
| BSF                                    | f, b | Bit Set f                    | 1        | 01            | 01bb | bfff | ffff   |          | 1, 2    |  |
| BTFSC                                  | f, b | Bit Test f, Skip if Clear    | 1 (2)    | 01            | 10bb | bfff | ffff   |          | 3       |  |
| BTFSS                                  | f, b | Bit Test f, Skip if Set      | 1 (2)    | 01            | 11bb | bfff | ffff   |          | 3       |  |
|                                        |      | LITERAL AND CONTROL          | OPERAT   | IONS          |      |      |        |          |         |  |
| ADDLW                                  | k    | Add literal and W            | 1        | 11            | 111x | kkkk | kkkk   | C, DC, Z |         |  |
| ANDLW                                  | k    | AND literal with W           | 1        | 11            | 1001 | kkkk | kkkk   | Z        |         |  |
| CALL                                   | k    | Call Subroutine              | 2        | 10            | 0kkk | kkkk | kkkk   |          |         |  |
| CLRWDT                                 | _    | Clear Watchdog Timer         | 1        | 00            | 0000 | 0110 | 0100   | TO, PD   |         |  |
| GOTO                                   | k    | Go to address                | 2        | 10            | 1kkk | kkkk | kkkk   |          |         |  |
| IORLW                                  | k    | Inclusive OR literal with W  | 1        | 11            | 1000 | kkkk | kkkk   | Z        |         |  |
| MOVLW                                  | k    | Move literal to W            | 1        | 11            |      | kkkk | kkkk   |          |         |  |
| RETFIE                                 | -    | Return from interrupt        | 2        | 00            | 0000 |      | 1001   |          |         |  |
| RETLW                                  | k    | Return with literal in W     | 2        | 11            | 01xx | kkkk | kkkk   |          |         |  |
| RETURN                                 | _    | Return from Subroutine       | 2        | 00            | 0000 | 0000 | 1000   |          |         |  |
| SLEEP                                  | _    | Go into Standby mode         | 1        | 00            | 0000 | 0110 | 0011   | TO, PD   |         |  |
| SUBLW                                  | k    | Subtract W from literal      | 1        | 11            | 110x | kkkk | kkkk   | C, DC, Z |         |  |
| XORLW                                  | k    | Exclusive OR literal with W  | 1        | 11            | 1010 | kkkk | kkkk   | Z        |         |  |

TABLE 2-2: PIC10(L)F320/322 MEMORY MAP (BANK 0)

| (*)                 | T   |           | 7   |                      | T   |                      | 1   |
|---------------------|-----|-----------|-----|----------------------|-----|----------------------|-----|
| INDF <sup>(*)</sup> | 00h | PMADRL    | 20h |                      | 40h |                      | 60h |
| TMR0                | 01h | PMADRH    | 21h |                      |     |                      |     |
| PCL                 | 02h | PMDATL    | 22h |                      |     |                      |     |
| STATUS              | 03h | PMDATH    | 23h |                      |     |                      |     |
| FSR                 | 04h | PMCON1    | 24h |                      |     |                      |     |
| PORTA               | 05h | PMCON2    | 25h |                      |     |                      |     |
| TRISA               | 06h | CLKRCON   | 26h |                      |     |                      |     |
| LATA                | 07h | NCO1ACCL  | 27h |                      |     |                      |     |
| ANSELA              | 08h | NCO1ACCH  | 28h |                      |     |                      |     |
| WPUA                | 09h | NCO1ACCU  | 29h |                      |     |                      |     |
| PCLATH              | 0Ah | NCO1INCL  | 2Ah |                      |     |                      |     |
| INTCON              | 0Bh | NCO1INCH  | 2Bh |                      |     |                      |     |
| PIR1                | 0Ch | Reserved  | 2Ch |                      |     |                      |     |
| PIE1                | 0Dh | NCO1CON   | 2Dh |                      |     |                      |     |
| OPTION_REG          | 0Eh | NCO1CLK   | 2Eh | General              |     | General              |     |
| PCON                | 0Fh | Reserved  | 2Fh | Purpose<br>Registers |     | Purpose<br>Registers |     |
| OSCCON              | 10h | WDTCON    | 30h | registers            |     | registers            |     |
| TMR2                | 11h | CLC1CON   | 31h | 32 Bytes             |     | 32 Bytes             |     |
| PR2                 | 12h | CLC1SEL1  | 32h |                      |     |                      |     |
| T2CON               | 13h | CLC1SEL2  | 33h |                      |     |                      |     |
| PWM1DCL             | 14h | CLC1POL   | 34h |                      |     |                      |     |
| PWM1DC              | 15h | CLC1GATE1 | 35h |                      |     |                      |     |
| PWM1CON             | 16h | CLC1GATE2 | 36h |                      |     |                      |     |
| PWM2DCL             | 17h | CLC1GATE3 | 37h |                      |     |                      |     |
| PWM2DC              | 18h | CLC1GATE4 | 38h |                      |     |                      |     |
| PWM2CON             | 19h | CWG1CON0  | 39h |                      |     |                      |     |
| IOCAP               | 1Ah | CWG1CON1  | 3Ah |                      |     |                      |     |
| IOCAN               | 1Bh | CWG1ASD   | 3Bh |                      |     |                      |     |
| IOCAF               | 1Ch | CWG1RC    | 3Ch |                      |     |                      |     |
| FVRCON              | 1Dh | CWG1FC    | 3Dh |                      |     |                      |     |
| ADRES               | 1Eh | VREGCON   | 3Eh |                      |     |                      |     |
| ADCON               | 1Fh | BORCON    | 3Fh |                      | 5Fh |                      | 7Fh |

**Legend:** = Unimplemented data memory locations, read as '0'.

<sup>\* =</sup> Not a physical register.

## Simplified PIC development steps

